home *** CD-ROM | disk | FTP | other *** search
/ Personal Computer World 2009 February / PCWFEB09.iso / Software / Linux / Kubuntu 8.10 / kubuntu-8.10-desktop-i386.iso / casper / filesystem.squashfs / usr / lib / python2.5 / tabnanny.pyc (.txt) < prev    next >
Python Compiled Bytecode  |  2008-10-29  |  8KB  |  283 lines

  1. # Source Generated with Decompyle++
  2. # File: in.pyc (Python 2.5)
  3.  
  4. '''The Tab Nanny despises ambiguous indentation.  She knows no mercy.
  5.  
  6. tabnanny -- Detection of ambiguous indentation
  7.  
  8. For the time being this module is intended to be called as a script.
  9. However it is possible to import it into an IDE and use the function
  10. check() described below.
  11.  
  12. Warning: The API provided by this module is likely to change in future
  13. releases; such changes may not be backward compatible.
  14. '''
  15. __version__ = '6'
  16. import os
  17. import sys
  18. import getopt
  19. import tokenize
  20. if not hasattr(tokenize, 'NL'):
  21.     raise ValueError("tokenize.NL doesn't exist -- tokenize module too old")
  22.  
  23. __all__ = [
  24.     'check',
  25.     'NannyNag',
  26.     'process_tokens']
  27. verbose = 0
  28. filename_only = 0
  29.  
  30. def errprint(*args):
  31.     sep = ''
  32.     for arg in args:
  33.         sys.stderr.write(sep + str(arg))
  34.         sep = ' '
  35.     
  36.     sys.stderr.write('\n')
  37.  
  38.  
  39. def main():
  40.     global filename_only, verbose
  41.     
  42.     try:
  43.         (opts, args) = getopt.getopt(sys.argv[1:], 'qv')
  44.     except getopt.error:
  45.         msg = None
  46.         errprint(msg)
  47.         return None
  48.  
  49.     for o, a in opts:
  50.         if o == '-q':
  51.             filename_only = filename_only + 1
  52.         
  53.         if o == '-v':
  54.             verbose = verbose + 1
  55.             continue
  56.     
  57.     if not args:
  58.         errprint('Usage:', sys.argv[0], '[-v] file_or_directory ...')
  59.         return None
  60.     
  61.     for arg in args:
  62.         check(arg)
  63.     
  64.  
  65.  
  66. class NannyNag(Exception):
  67.     '''
  68.     Raised by tokeneater() if detecting an ambiguous indent.
  69.     Captured and handled in check().
  70.     '''
  71.     
  72.     def __init__(self, lineno, msg, line):
  73.         self.lineno = lineno
  74.         self.msg = msg
  75.         self.line = line
  76.  
  77.     
  78.     def get_lineno(self):
  79.         return self.lineno
  80.  
  81.     
  82.     def get_msg(self):
  83.         return self.msg
  84.  
  85.     
  86.     def get_line(self):
  87.         return self.line
  88.  
  89.  
  90.  
  91. def check(file):
  92.     '''check(file_or_dir)
  93.  
  94.     If file_or_dir is a directory and not a symbolic link, then recursively
  95.     descend the directory tree named by file_or_dir, checking all .py files
  96.     along the way. If file_or_dir is an ordinary Python source file, it is
  97.     checked for whitespace related problems. The diagnostic messages are
  98.     written to standard output using the print statement.
  99.     '''
  100.     if os.path.isdir(file) and not os.path.islink(file):
  101.         if verbose:
  102.             print '%r: listing directory' % (file,)
  103.         
  104.         names = os.listdir(file)
  105.         for name in names:
  106.             fullname = os.path.join(file, name)
  107.             if os.path.isdir(fullname) or not os.path.islink(fullname) or os.path.normcase(name[-3:]) == '.py':
  108.                 check(fullname)
  109.                 continue
  110.         
  111.         return None
  112.     
  113.     
  114.     try:
  115.         f = open(file)
  116.     except IOError:
  117.         msg = None
  118.         errprint('%r: I/O Error: %s' % (file, msg))
  119.         return None
  120.  
  121.     if verbose > 1:
  122.         print 'checking %r ...' % file
  123.     
  124.     
  125.     try:
  126.         process_tokens(tokenize.generate_tokens(f.readline))
  127.     except tokenize.TokenError:
  128.         msg = None
  129.         errprint('%r: Token Error: %s' % (file, msg))
  130.         return None
  131.     except IndentationError:
  132.         msg = None
  133.         errprint('%r: Indentation Error: %s' % (file, msg))
  134.         return None
  135.     except NannyNag:
  136.         nag = None
  137.         badline = nag.get_lineno()
  138.         line = nag.get_line()
  139.         if verbose:
  140.             print '%r: *** Line %d: trouble in tab city! ***' % (file, badline)
  141.             print 'offending line: %r' % (line,)
  142.             print nag.get_msg()
  143.         elif ' ' in file:
  144.             file = '"' + file + '"'
  145.         
  146.         if filename_only:
  147.             print file
  148.         else:
  149.             print file, badline, repr(line)
  150.         return None
  151.  
  152.     if verbose:
  153.         print '%r: Clean bill of health.' % (file,)
  154.     
  155.  
  156.  
  157. class Whitespace:
  158.     (S, T) = ' \t'
  159.     
  160.     def __init__(self, ws):
  161.         self.raw = ws
  162.         S = Whitespace.S
  163.         T = Whitespace.T
  164.         count = []
  165.         b = n = nt = 0
  166.         for ch in self.raw:
  167.             if ch == S:
  168.                 n = n + 1
  169.                 b = b + 1
  170.                 continue
  171.             if ch == T:
  172.                 n = n + 1
  173.                 nt = nt + 1
  174.                 if b >= len(count):
  175.                     count = count + [
  176.                         0] * ((b - len(count)) + 1)
  177.                 
  178.                 count[b] = count[b] + 1
  179.                 b = 0
  180.                 continue
  181.             break
  182.         
  183.         self.n = n
  184.         self.nt = nt
  185.         self.norm = (tuple(count), b)
  186.         self.is_simple = len(count) <= 1
  187.  
  188.     
  189.     def longest_run_of_spaces(self):
  190.         (count, trailing) = self.norm
  191.         return max(len(count) - 1, trailing)
  192.  
  193.     
  194.     def indent_level(self, tabsize):
  195.         (count, trailing) = self.norm
  196.         il = 0
  197.         for i in range(tabsize, len(count)):
  198.             il = il + (i / tabsize) * count[i]
  199.         
  200.         return trailing + tabsize * (il + self.nt)
  201.  
  202.     
  203.     def equal(self, other):
  204.         return self.norm == other.norm
  205.  
  206.     
  207.     def not_equal_witness(self, other):
  208.         n = max(self.longest_run_of_spaces(), other.longest_run_of_spaces()) + 1
  209.         a = []
  210.         for ts in range(1, n + 1):
  211.             if self.indent_level(ts) != other.indent_level(ts):
  212.                 a.append((ts, self.indent_level(ts), other.indent_level(ts)))
  213.                 continue
  214.         
  215.         return a
  216.  
  217.     
  218.     def less(self, other):
  219.         if self.n >= other.n:
  220.             return False
  221.         
  222.         if self.is_simple and other.is_simple:
  223.             return self.nt <= other.nt
  224.         
  225.         n = max(self.longest_run_of_spaces(), other.longest_run_of_spaces()) + 1
  226.         for ts in range(2, n + 1):
  227.             if self.indent_level(ts) >= other.indent_level(ts):
  228.                 return False
  229.                 continue
  230.         
  231.         return True
  232.  
  233.     
  234.     def not_less_witness(self, other):
  235.         n = max(self.longest_run_of_spaces(), other.longest_run_of_spaces()) + 1
  236.         a = []
  237.         for ts in range(1, n + 1):
  238.             if self.indent_level(ts) >= other.indent_level(ts):
  239.                 a.append((ts, self.indent_level(ts), other.indent_level(ts)))
  240.                 continue
  241.         
  242.         return a
  243.  
  244.  
  245.  
  246. def format_witnesses(w):
  247.     firsts = map((lambda tup: str(tup[0])), w)
  248.     prefix = 'at tab size'
  249.     if len(w) > 1:
  250.         prefix = prefix + 's'
  251.     
  252.     return prefix + ' ' + ', '.join(firsts)
  253.  
  254.  
  255. def process_tokens(tokens):
  256.     INDENT = tokenize.INDENT
  257.     DEDENT = tokenize.DEDENT
  258.     NEWLINE = tokenize.NEWLINE
  259.     JUNK = (tokenize.COMMENT, tokenize.NL)
  260.     indents = [
  261.         Whitespace('')]
  262.     check_equal = 0
  263.     for type, token, start, end, line in tokens:
  264.         if type == NEWLINE:
  265.             check_equal = 1
  266.             continue
  267.         if type == INDENT:
  268.             check_equal = 0
  269.             thisguy = Whitespace(token)
  270.             if not indents[-1].less(thisguy):
  271.                 witness = indents[-1].not_less_witness(thisguy)
  272.                 msg = 'indent not greater e.g. ' + format_witnesses(witness)
  273.                 raise NannyNag(start[0], msg, line)
  274.             
  275.             indents.append(thisguy)
  276.             continue
  277.         None if type == DEDENT else indents[-1].equal(thisguy)
  278.     
  279.  
  280. if __name__ == '__main__':
  281.     main()
  282.  
  283.